home *** CD-ROM | disk | FTP | other *** search
- /* env.c, envread.c, env.h: environ library
- Daniel J. Bernstein, djb@silverton.berkeley.edu.
- Depends on str.h, alloc.h.
- Requires environ.
- 19960113: rewrite. warning: interface is different.
- No known patent problems.
- */
-
- #include "str.h"
- #include "alloc.h"
- #include "env.h"
-
- int env_isinit = 0; /* if env_isinit: */
- static int ea; /* environ is a pointer to ea+1 char*'s. */
- static int en; /* the first en of those are ALLOCATED. environ[en] is 0. */
-
- static void env_goodbye(i) int i;
- {
- alloc_free(environ[i]);
- environ[i] = environ[--en];
- environ[en] = 0;
- }
-
- static char *null = 0;
-
- void env_clear()
- {
- if (env_isinit) while (en) env_goodbye(0);
- else environ = &null;
- }
-
- static void env_unsetlen(s,len) char *s; int len;
- {
- int i;
- for (i = en - 1;i >= 0;--i)
- if (!str_diffn(s,environ[i],len))
- if (environ[i][len] == '=')
- env_goodbye(i);
- }
-
- int env_unset(s) char *s;
- {
- if (!env_isinit) if (!env_init()) return 0;
- env_unsetlen(s,str_len(s));
- return 1;
- }
-
- static int env_add(s) char *s;
- {
- char *t;
- t = env_findeq(s);
- if (t) env_unsetlen(s,t - s);
- if (en == ea)
- {
- ea += 30;
- if (!alloc_re(&environ,(en + 1) * sizeof(char *),(ea + 1) * sizeof(char *)))
- { ea = en; return 0; }
- }
- environ[en++] = s;
- environ[en] = 0;
- return 1;
- }
-
- int env_put(s) char *s;
- {
- char *u;
- if (!env_isinit) if (!env_init()) return 0;
- u = alloc(str_len(s) + 1);
- if (!u) return 0;
- str_copy(u,s);
- if (!env_add(u)) { alloc_free(u); return 0; }
- return 1;
- }
-
- int env_put2(s,t) char *s; char *t;
- {
- char *u;
- int slen;
- if (!env_isinit) if (!env_init()) return 0;
- slen = str_len(s);
- u = alloc(slen + str_len(t) + 2);
- if (!u) return 0;
- str_copy(u,s);
- u[slen] = '=';
- str_copy(u + slen + 1,t);
- if (!env_add(u)) { alloc_free(u); return 0; }
- return 1;
- }
-
- int env_init()
- {
- char **newenviron;
- int i;
- for (en = 0;environ[en];++en) ;
- ea = en + 10;
- newenviron = (char **) alloc((ea + 1) * sizeof(char *));
- if (!newenviron) return 0;
- for (en = 0;environ[en];++en)
- {
- newenviron[en] = alloc(str_len(environ[en]) + 1);
- if (!newenviron[en])
- {
- for (i = 0;i < en;++i) alloc_free(newenviron[i]);
- alloc_free(newenviron);
- return 0;
- }
- str_copy(newenviron[en],environ[en]);
- }
- newenviron[en] = 0;
- environ = newenviron;
- env_isinit = 1;
- return 1;
- }
-